Map是一个集合类的接口。
HashMap
它实现了Map接口,存储键值对。底层实现是散列表,即数组加链表的结构。有几个重要的参数:
Size 当前Hashmap存储元素的个数
Capacity 容量,当前Hashmap最多能够发现多少元素
Load factor,决定什么时候对Hashmap进行拓容,当Size/Capacity > Load factor时,进行拓容。
put
对key的hashCode()做hash,然后再计算index;
如果没碰撞直接放到bucket里;
如果碰撞了,以链表的形式存在buckets后;
如果碰撞导致链表过长(大于等于TREEIFY_THRESHOLD),就把链表转换成红黑树;(在JDK8里,新增默认为8的閥值[TREEIFY_THRESHOLD],当一个桶里的Entry超过閥值,就不以单向链表而以红黑树来存放以加快Key的查找速度)
如果节点已经存在就替换old value(保证key的唯一性)
如果bucket满了(超过load factor*current capacity),就要resize
get
bucket里的第一个节点,直接命中;
如果有冲突,则通过key.equals(k)去查找对应的entry
若为树,则在树中通过key.equals(k)查找,O(logn);
若为链表,则在链表中通过key.equals(k)查找,O(n)。
hash函数
|
|
n其实就是capacity,数组的元素个数,这里保证它是的2的n次方,使用与操作比求余操作性能更好
TreeMap
同样是存储键值对,TreeMap
底层使用了红黑树,最大的特点就是它能够基于键值的自然顺序进行拍讯。使得树具有不错的平衡性,这样操作的速度就可以达到log(n)的水平了。TreeMap的基本操作 containsKey、get、put 和 remove 的时间复杂度是 log(n)
LinkedHashMap
LinkedHashMap是Hash表和链表的实现,并且依靠着双向链表保证了迭代顺序是插入的顺序
ConcurrentHashMap
以上的Map都是非线程安全的,当设计多线程时,需要手动实现互斥同步,而ConcurrentHashMap从名字上来看就知道它能够实现线程安全的Map
它的原理也是散列表,只不过进行了封装,新增了Segment。它的结构包含Segment 的数组,在默认的并发级别会创建包含 16 个 Segment 对象的数组。通过我们上面的知识,我们知道每个 Segment 又包含若干个散列表的桶,每个桶是由 HashEntry 链接起来的一个链表。
因此,在不同Segment里操作数据,可以并发,不用加锁,而在相同的Segment,自动实现了互斥同步。
WeakHashmap
以弱键 实现的基于哈希表的 Map。在 WeakHashMap 中,当某个键不再正常使用时,将自动移除其条目。更精确地说,对于一个给定的键,其映射的存在并不阻止垃圾回收器对该键的丢弃,这就使该键成为可终止的,被终止,然后被回收。WeekHashMap 的这个特点特别适用于需要缓存的场景。